001    /*
002     * Copyright 2005 Stephen J. McConnell.
003     *
004     * Licensed  under the  Apache License,  Version 2.0  (the "License");
005     * you may not use  this file  except in  compliance with the License.
006     * You may obtain a copy of the License at
007     *
008     *   http://www.apache.org/licenses/LICENSE-2.0
009     *
010     * Unless required by applicable law or agreed to in writing, software
011     * distributed  under the  License is distributed on an "AS IS" BASIS,
012     * WITHOUT  WARRANTIES OR CONDITIONS  OF ANY KIND, either  express  or
013     * implied.
014     *
015     * See the License for the specific language governing permissions and
016     * limitations under the License.
017     */
018    
019    package net.dpml.station.server; 
020    
021    import java.rmi.AlreadyBoundException;
022    import java.rmi.RemoteException;
023    import java.rmi.server.UnicastRemoteObject;
024    import java.rmi.registry.LocateRegistry;
025    import java.rmi.registry.Registry;
026    import java.net.URL;
027    import java.util.Map;
028    import java.util.Hashtable;
029    import java.util.LinkedList;
030    import java.util.List;
031    import java.util.EventObject;
032    
033    import net.dpml.station.Application;
034    import net.dpml.station.Callback;
035    import net.dpml.station.Manager;
036    import net.dpml.station.Station;
037    import net.dpml.station.StationException;
038    
039    import net.dpml.station.info.ApplicationDescriptor;
040    import net.dpml.station.info.StartupPolicy;
041    import net.dpml.station.ApplicationRegistry;
042    
043    import net.dpml.util.Logger;
044    import net.dpml.transit.model.TransitModel;
045    import net.dpml.transit.Disposable;
046    
047    import net.dpml.lang.UnknownKeyException;
048    
049    /**
050     * The RemoteStation is responsible for the establishment of 
051     * callback monitors to external processes established by the 
052     * station manager.
053     * @author <a href="http://www.dpml.net">Digital Product Meta Library</a>
054     * @version 1.0.0
055     */
056    public class RemoteStation extends UnicastRemoteObject implements Station, Manager
057    {
058        private final RemoteApplicationRegistry m_registry;
059        private final Map m_applications = new Hashtable();
060        private final Logger m_logger;
061        private final int m_port;
062        private final Registry m_rmiRegistry;
063        private final URL m_store;
064        private final TransitModel m_model;
065        private final LoggingServer m_server;
066        private final Thread m_thread;
067        
068        private boolean m_terminated = false;
069        
070       /**
071        * Creation of a station instance.
072        *
073        * @param logger the assigned logging channel
074        * @param model the transit model
075        * @param port the station port 
076        * @param registryStorageUrl uri defining the registry backing store
077        * @exception Exception if a exception occurs during establishment
078        */
079        public RemoteStation( 
080          Logger logger, TransitModel model, int port, URL registryStorageUrl ) 
081          throws Exception
082        {
083            super();
084            
085            m_logger = logger;
086            m_port = port;
087            m_store = registryStorageUrl;
088            m_model = model;
089            
090            m_rmiRegistry = getLocalRegistry( port );
091            
092            try
093            {
094                m_rmiRegistry.bind( STATION_KEY, this );
095            }
096            catch( AlreadyBoundException e )
097            {
098                final String error =
099                 "An instance of the Station is already bound to port " + port;
100                throw new StationException( error, e );
101            }
102    
103            setShutdownHook( this );
104            startEventDispatchThread();
105    
106            try
107            {
108                m_server = new LoggingServer( 2020 );
109                m_thread = new Thread( m_server, "DPML Station Logging Server" );
110                m_thread.start();
111            }
112            catch( Exception e )
113            {
114                final String error =
115                 "Unexpected error while attempting to start the logging server on port " + 2020;
116                throw new StationException( error, e );
117            }
118            
119            if( getLogger().isDebugEnabled() )
120            {
121                if( null == registryStorageUrl )
122                {
123                    getLogger().debug( "loading registry from default storage" );
124                }
125                else
126                {
127                    getLogger().debug( "loading registry from [" + registryStorageUrl + "]" );
128                }
129            }
130            
131            m_registry = new RemoteApplicationRegistry( logger, registryStorageUrl );
132            String[] keys = m_registry.getKeys();
133            
134            if( getLogger().isDebugEnabled() )
135            {
136                getLogger().debug( "registry established (" + keys.length + ")" );
137            }
138            
139            for( int i=0; i<keys.length; i++ )
140            {
141                String key = keys[i];
142                try
143                {
144                    ApplicationDescriptor descriptor = 
145                      m_registry.getApplicationDescriptor( key );
146                    if( StartupPolicy.AUTOMATIC.equals( descriptor.getStartupPolicy() ) )
147                    {
148                        RemoteApplication application = getRemoteApplication( key );
149                        application.start();
150                    }
151                }
152                catch( UnknownKeyException e )
153                {
154                    throw new RuntimeException( e ); // will not happen
155                }
156            }
157        }
158        
159       /**
160        * Return a string containing info about the general setup of the station.
161        * @return station configuration info
162        */
163        public String[] getInfo()
164        {
165            String[] values = new String[4];
166            values[0] = "Port: " + m_port;
167            values[1] = "Store: " + m_store;
168            values[2] = "Basedir: " + System.getProperty( "user.dir" );
169            values[3] = "Codebase: " 
170              + getClass().getProtectionDomain().getCodeSource().getLocation();
171            return values;
172        }
173        
174       /**
175        * Return an callback handler for the supplied id.
176        * @param id the callback id
177        * @return the callback handler
178        * @exception UnknownKeyException if the id is unknown
179        * @exception RemoteException if a remote error occurs
180        */
181        public Callback getCallback( String id ) throws UnknownKeyException, RemoteException
182        {
183            // TODO: improve this so that this is only called once per appliation
184            return getRemoteApplication( id );
185        }
186        
187       /**
188        * Shutdown the station.
189        */
190        public void shutdown()
191        {
192            shutdown( true );
193        }
194        
195       /**
196        * Shutdown the station.
197        * @param exit if true launch a process termination
198        */
199        private void shutdown( boolean exit )
200        {
201            synchronized( m_applications )
202            {
203                if( m_terminated )
204                {
205                    return;
206                }
207                else
208                {
209                    m_terminated = true;
210                }
211                
212                if( getLogger().isInfoEnabled() )
213                {
214                    getLogger().info( "initiating station shutdown" );
215                }
216                
217                try
218                {
219                    m_rmiRegistry.unbind( STATION_KEY );
220                }
221                catch( Exception e )
222                {
223                    // ignore
224                }
225                try
226                {
227                    RemoteApplication[] applications = getRemoteApplications();
228                    for( int i=0; i<applications.length; i++ )
229                    {
230                        RemoteApplication application = applications[i];
231                        application.shutdown();
232                        UnicastRemoteObject.unexportObject( application, true );
233                    }
234                    UnicastRemoteObject.unexportObject( m_registry, true );
235                }
236                catch( Exception e )
237                {
238                    // ignore
239                }
240                try
241                {
242                    int n =  m_server.getErrorCount();
243                    if( n > 0 )
244                    {
245                        getLogger().warn( "logging issues: " + n );
246                    }
247                    m_thread.interrupt();
248                }
249                catch( Exception e )
250                {
251                    // ignore
252                }
253                
254                finally
255                {
256                    if( getLogger().isInfoEnabled() )
257                    {
258                        getLogger().info( "station shutdown complete" );
259                    }
260                    
261                    if( exit )
262                    {
263                        if( m_model instanceof Disposable )
264                        {
265                            try
266                            {
267                                Disposable disposable = (Disposable) m_model;
268                                disposable.dispose();
269                            }
270                            catch( Exception e )
271                            {
272                                // ignore
273                            }
274                        }
275                        
276                        if( getLogger().isDebugEnabled() )
277                        {
278                            getLogger().debug( "terminating process" );
279                        }
280                        
281                        Thread thread = new Thread(
282                          new Runnable()
283                          {
284                            public void run()
285                            {
286                                RemoteStation.m_DISPATCH.dispose();
287                                System.exit( 0 );
288                            }
289                          }
290                        );
291                        thread.start();
292                    }
293                }
294            }
295        }
296        
297       /**
298        * Return the application registry.
299        * @return the registry
300        */
301        public ApplicationRegistry getApplicationRegistry()
302        {
303            return m_registry;
304        }
305    
306       /**
307        * Return an application reference for the supplied key.
308        * @param key the application key
309        * @return the application
310        * @exception UnknownKeyException if the key is unknown
311        * @exception RemoteException if a remote error occurs
312        */
313        public Application getApplication( String key ) throws UnknownKeyException, RemoteException
314        {
315            return getRemoteApplication( key );
316        }
317        
318       /**
319        * Return an application reference for the supplied key.
320        * @param key the application key
321        * @return the application
322        * @exception UnknownKeyException if the key is unknown
323        * @exception RemoteException if a remote error occurs
324        */
325        RemoteApplication getRemoteApplication( String key ) throws UnknownKeyException, RemoteException
326        {
327            synchronized( m_applications )
328            {
329                if( m_applications.containsKey( key ) )
330                {
331                    return (RemoteApplication) m_applications.get( key );
332                }
333                else
334                {
335                    Logger logger = getLogger().getChildLogger( key );
336                    ApplicationDescriptor descriptor = m_registry.getApplicationDescriptor( key );
337                    RemoteApplication application = 
338                      new RemoteApplication( logger, descriptor, key, m_port );
339                    m_applications.put( key, application );
340                    return application;
341                }
342            }
343        }
344    
345       /**
346        * Return an array of all remote applications.
347        * @return the applications array
348        */
349        RemoteApplication[] getRemoteApplications()
350        {
351            synchronized( m_applications )
352            {
353                return (RemoteApplication[]) m_applications.values().toArray( new RemoteApplication[0] );
354            }
355        }
356        
357        private Logger getLogger()
358        {
359            return m_logger;
360        }
361        
362        private Registry getLocalRegistry( int port ) throws RemoteException
363        {
364            try
365            {
366                Registry registry = LocateRegistry.createRegistry( port );
367                getLogger().debug( "created local registry on port " + port );
368                return registry;
369            }
370            catch( RemoteException e )
371            {
372                Registry registry = LocateRegistry.getRegistry( port );
373                getLogger().debug( "using local registry on port " + port );
374                return registry;
375            }
376        }
377    
378        /**
379         * Queue of pending notification events.  When an event for which 
380         * there are one or more listeners occurs, it is placed on this queue 
381         * and the queue is notified.  A background thread waits on this queue 
382         * and delivers the events.  This decouples event delivery from 
383         * the application concern, greatly simplifying locking and reducing 
384         * opportunity for deadlock.
385         */
386        private static final List EVENT_QUEUE = new LinkedList();
387    
388       /**
389        * Enqueue an event for delivery to registered
390        * listeners unless there are no registered
391        * listeners.
392        * @param event the event to enqueue
393        */
394        static void enqueueEvent( EventObject event )
395        {
396            synchronized( EVENT_QUEUE ) 
397            {
398                EVENT_QUEUE.add( event );
399                EVENT_QUEUE.notify();
400            }
401        }
402        
403        /**
404         * A single background thread ("the event notification thread") monitors
405         * the event queue and delivers events that are placed on the queue.
406         */
407        private static class EventDispatchThread extends Thread 
408        {
409            private final Logger m_logger;
410            
411            private boolean m_continue = true;
412            
413            EventDispatchThread( Logger logger )
414            {
415                super( "DPML Station Event Dispatch" );
416                m_logger = logger;
417                m_logger.debug( "starting event dispatch thread" );
418            }
419            
420            void dispose()
421            {
422                synchronized( EVENT_QUEUE )
423                {
424                    m_logger.debug( "stopping event dispatch thread" );
425                    m_continue = false;
426                    EVENT_QUEUE.notify();
427                }
428            }
429            
430            public void run() 
431            {
432                while( m_continue ) 
433                {
434                    // Wait on EVENT_QUEUE till an event is present
435                    EventObject event = null;
436                    synchronized( EVENT_QUEUE ) 
437                    {
438                        try
439                        {
440                            while( EVENT_QUEUE.isEmpty() )
441                            {
442                                EVENT_QUEUE.wait();
443                            }
444                            Object object = EVENT_QUEUE.remove( 0 );
445                            try
446                            {
447                                event = (EventObject) object;
448                            }
449                            catch( ClassCastException cce )
450                            {
451                                final String error = 
452                                  "Unexpected class cast exception while processing an event." 
453                                  + "\nEvent: " + object;
454                                throw new IllegalStateException( error );
455                            }
456                        }
457                        catch( InterruptedException e )
458                        {
459                            return;
460                        }
461                    }
462                    
463                    Object source = event.getSource();
464                    if( source instanceof UnicastEventSource )
465                    {
466                        UnicastEventSource producer = (UnicastEventSource) source;
467                        try
468                        {
469                            producer.processEvent( event );
470                        }
471                        catch( Throwable e )
472                        {
473                            final String error = 
474                              "Unexpected error while processing event."
475                              + "\nEvent: " + event
476                              + "\nSource: " + source;
477                            m_logger.warn( error, e );
478                        }
479                    }
480                    else
481                    {
482                        final String error = 
483                          "Event source [" 
484                          + source.getClass().getName()
485                          + "] is not an instance of " + UnicastEventSource.class.getName();
486                        throw new IllegalStateException( error );
487                    }
488                }
489                
490                m_logger.info( "Controller event queue terminating." );
491            }
492        }
493    
494        private static EventDispatchThread m_DISPATCH = null;
495    
496        /**
497         * This method starts the event dispatch thread the first time it
498         * is called.  The event dispatch thread will be started only
499         * if someone registers a listener.
500         */
501        private synchronized void startEventDispatchThread()
502        {
503            if( m_DISPATCH == null )
504            {
505                Logger logger = getLogger();
506                m_DISPATCH = new EventDispatchThread( logger );
507                m_DISPATCH.setDaemon( true );
508                m_DISPATCH.start();
509            }
510        }
511            
512       /**
513        * Create a shutdown hook that will trigger shutdown of the supplied plugin.
514        * @param station the station
515        */
516        public static void setShutdownHook( final RemoteStation station )
517        {
518            //
519            // Create a shutdown hook to trigger clean disposal of the
520            // controller
521            //
522            
523            Runtime.getRuntime().addShutdownHook(
524              new Thread()
525              {
526                  public void run()
527                  {
528                      try
529                      {
530                          station.shutdown();
531                      }
532                      catch( Throwable e )
533                      {
534                          System.err.println( e.toString() );
535                      }
536                      System.runFinalization();
537                  }
538              }
539            );
540        }
541        
542    }